library(ISLR)
<- College college_data
Section 9 - Support Vector Machines
Notes
Note that the material in these notes draws on the excellent past notes by TFs Laura Morris, Emily Mower and more thorough treatment of these topics in by Gareth James, Daniela Witten, Trevor Hastie and Robert Tibshirani.
Support Vector Machines
In this class, we will see a series of methods that fall under the support vector umbrella but which vary significantly in their flexibility. We start with maximal margin classifiers, which are the most rigid and only work when the data can be perfectly separated with a linear decision boundary. Then, we will cover support vector classifiers, which still uses a linear decision boundary, but which can accommodate data that cannot be perfectly separated by such a decision boundary. We will then cover support vector machines, which flexibly transform the original data to allow for decision boundaries that are non-linear in the original feature space (though they remain linear in the transformed feature space). Like support vector classifiers, support vector machines will also accommodate data that is not perfectly separable. Finally, we will touch upon extending the support vector ideas to multiclass settings.
Maximal Margin Classifier
Maximal margin classifiers use a separating hyperplane to divide the feature space in two, with the idea being that all observations in one class lie on one side of the separating hyperplane while all observations of the other class lie on the other side.
Separating Hyperplanes
A hyperplane is a flat affine subspace that has one fewer dimensions than the feature space. The first part of the definition (“flat affine subspace”) means that the subspace can be described by a linear equation and does not need to pass through the origin. The second part of the definition (about the dimensionality) means that with
Imagine a dataset with only two features,
The separating hyperplane that divides the feature space with
Classifications with Separating Hyperplanes
We can then use this equation as our classifier. Specifically, let
If the resulting value is negative, the point lies to one side of the hyperplane and is assigned to class
Note that we are now using
Maximal Margin Classifier
The maximal margin classifier can be used when the data is perfectly separable, meaning that there exists a separating hyperplane such that all observations from one class fall on one side of the hyperplane and all observations from the other class fall on the other side of the hyperplane. However, when one such separating hyperplane exists, usually infinite such separating hyperplanes exist. The maximal margin classifier provides a disciplined way to choose among them.
Specifically, it measures the distance between each point and the separating hyperplane. The minimum distance between all points and the separating hyperplane is referred to as the margin. The maximal margin classifier selects that separating hyperplane that leads to the largest possible margin, that is, the largest possible distance between the separating hyperplane and the closest training points. Note that as you increase the distance between the decision boundary and the closest point from one class, you must decrease the distance between the decision boundary and the closest point(s) from the other class. Therefore, the margin will always be defined by at least one point from each class. These points are called ``support’’ points or vectors and the model is fully defined by them, so if they move the model changes. Other points can move however they like outside the margins, and as long as they do not cross the margins, the model will not change.
When the margin is large, it suggests the classes are very well separated and performance on a test set should be good. When the margin is small, the classes are only barely separated and the exact decision boundary may not do as well in the test set.
Support Vector Classifier
Sometimes the data cannot be perfectly separated by a separating hyperplane. In such cases, the support vector classifier offers a generalization of the maximal margin classifier.
The support vector classifier is extremely similar to the maximal margin classifier, except that it allows some points to cross the margin and even the decision boundary (e.g. be misclassified). Because some points can cross the margin under the support vector classifier, we call the margin a “soft margin” compared to the “hard margin” of the maximal margin classifier, which does not permit any points to cross it.
The support vector classifier improves empirically relative to the maximal margin classifier in three key ways. One, it works for data that cannot be perfectly separated by a hyperplane. Two, it is more robust to individual points, meaning that because it allows points to cross the margin, movement in points near the margin will not have the same dramatic impact on the margin and decision boundary that they would have had under the maximal margin classifier. Third, related to this point, the support vector classifier yields better predictions for most training observations.
The support vector classifier works by creating slack variables
The budget
In the support vector classifier, only observations that lie on the margin or that violate the margin will affect the hyperplane. Just as with the maximal margin classifier, points that lie on the correct side of the margin will not be used to define the hyperplane. As
Support Vector Machine
The support vector machine extends the support vector classifier to the case where the decision boundary is non-linear. It is somewhat analogous to using polynomial regression when the linearity assumption of linear regression does not hold. Specifically, it works with a transformed feature space and finds a decision boundary that is linear in the transformed space, but which is non-linear in the original space.
However, the way in which the support vector machine transforms the original space is new. It does the transformation using a kernel
The values of
The argument for why we use a kernel rather than an enlarged feature space has to do with computational efficiency. Using the kernel only requires computing the kernel (the generalized inner product) for each unique pair $(x_i, x_{i^}) $ in the training data; it does not require explicitly working in a transformed feature space, which may be computationally intractable. For example, suppose we wanted to expand our feature space from
to
The kernel function can take many forms, including polynomial and radial functions. Different kernel functions allow for different levels of model flexibility. However, note that as with many other models we’ve seen this semester, the more flexible we make the kernel, the more likely we are to fit the training data well (low bias) but risk overfitting the training data (high variance). Therefore, in the support vector machine, a very flexible kernel function leads to low bias but high variance.
Extensions to Multiclass Problems
The idea of support vector machines does not generalize easily to the multiclass setting, but two options have been proposed.
One is called the one-versus-one approach, where a collection of models is built that each evaluate the question of whether the observation belongs to class
The other is called the one-versus-all approach, where
Code
This week, we will use the College data that is part of the ISLR package
We will start by fitting a support vector classifier to the College data. The College data has 18 variables. We will treat “Private” as the outcome variable and the acceptance rates and out-of-state tuition as features. Since the first variable currently exists as a count (not rate), we begin by creating the new variable. To do this, we will use the dplyr package. Note: if your code won’t run, make sure your MASS package isn’t loaded.
library(dplyr)
<- college_data %>%
transformed_data mutate(accept_rate = Accept/Apps) %>%
select(accept_rate, Outstate, Private)
The svm function will have a hard time unless we scale our data. There is an option to scale within the svm()
function that is perfectly sufficient when training a model. However, we will be doing a lot of visual inspection, so it will help to have our raw data scaled for these exercises.
<- transformed_data %>%
transformed_data mutate(accept_rate = scale(accept_rate),
Outstate = scale(Outstate))
To help with visual interpretation, we will take a random subset of the full data to use in most examples
set.seed(222)
<- transformed_data[sample(1:nrow(transformed_data), 100),] small_data
Later, it will be convenient to have the features in their own matrix and the outcome in its own vector, so we do that now
<- as.matrix(small_data[, 1:2])
x <- if_else(small_data[, 3] == "Yes", 1, -1)
y summary(transformed_data)
accept_rate.V1 Outstate.V1 Private
Min. :-4.027367 Min. :-2.0135809 No :212
1st Qu.:-0.484557 1st Qu.:-0.7757038 Yes:565
Median : 0.216325 Median :-0.1120227
Mean : 0.000000 Mean : 0.0000000
3rd Qu.: 0.690631 3rd Qu.: 0.6175294
Max. : 1.720364 Max. : 2.7987285
We will start by looking at our data, where the color represents whether the college is private or public
plot(small_data[, c("accept_rate", "Outstate")],
col = small_data$Private)
Clearly, our data is not perfectly separable, so we cannot use the maximal margin classifier. Let’s try a support vector classifier. For this, we use the package e1071, which implements support vector machines in R. Recall that the support vector classifier is a special case of the support vector machine where the kernel is linear. We have been talking about a “budget” for points that violate the margin or decision boundary. This package uses “cost”, which is basically the inverse – how much should an error cost? If we want to allow a big budget, that translates to using a small cost parameter. The default cost is 1.
# install.packages("e1071")
library(e1071)
<- svm(Private ~ Outstate + accept_rate,
college_svm1 data = small_data,
kernel = "linear",
cost = 10,
scale = FALSE)
plot(college_svm1, small_data)
There are a few main things we can see from this plot:
- The decision boundary, which is linear even though it looks jagged
- The support vectors (x’s) and all other points (o’s)
- The regions where the model predicts “Yes” and “No”
We can also learn about our model using summary()
summary(college_svm1)
Call:
svm(formula = Private ~ Outstate + accept_rate, data = small_data,
kernel = "linear", cost = 10, scale = FALSE)
Parameters:
SVM-Type: C-classification
SVM-Kernel: linear
cost: 10
Number of Support Vectors: 32
( 16 16 )
Number of Classes: 2
Levels:
No Yes
From this output, we see: - (1) The number of support vectors, total and in each class - (2) The type of kernel (“linear”) and cost (10)
We can find which points serve as support vectors by getting their row numbers. Note that the “Yes” SV indices are listed first, followed by the “No” SV indices
$index college_svm1
[1] 8 12 15 21 24 31 32 39 50 52 56 61 62 65 92 94 4 6 7 10 17 25 40 70 71
[26] 72 74 75 87 88 95 98
The svm() function in R does not immediately return the coefficients for the support vector classifier. It is believed that the reason for this is that the function is a very general function and the linear kernel used for the support vector classifier is only one of the many kernels it can handle. However, thanks to code from Data Camp, we can recover the the coefficients. Note that rho actually gives -beta0 “t” in line 100 below stands for transpose, “drop” get rid off redundant dimensions. See link below for more information https://statisticsglobe.com/drop-function-r
<- drop(t(college_svm1$coefs) %*% x[college_svm1$index,])
beta <- -college_svm1$rho
beta0
print(beta)
accept_rate Outstate
0.7777781 2.6005940
print(beta0)
[1] 1.749692
To understand our model better, we will:
- Create a grid of x-values that cover our feature space
- Predict a class for each point on the grid to get shading
- Plot our points and highlight support vectors
- Plot the decision boundary and margins
For 1, I will us a function provided by Data Camp:
= function(x, n = 75) {
make.grid <- apply(x, 2, range)
grange <- seq(from = grange[1, 1], to = grange[2, 1], length = n)
x1 <- seq(from = grange[1, 2], to = grange[2, 2], length = n)
x2 expand.grid(X1 = x1, X2 = x2)
}
<- make.grid(x)
xgrid colnames(xgrid) <- colnames(x)
For 2, we will use the predict() function to get the class for each point on the grid
<- predict(college_svm1, xgrid) ygrid
For 3, we will plot the grid, then the points, then the support vectors
plot(xgrid,
col = c("red","blue")[as.numeric(ygrid)],
pch = 20,
cex = .2)
points(x,
col = y + 3,
pch = 19)
points(x[college_svm1$index,],
pch = 5,
cex = 2)
For (4), we can use the function abline, which takes the intercept and slope of a line. Recall that the coefficients we have recovered are of the form b0 + b1 * x1 + b2 * x2 = 0. To convert that to the abline form, we do some elementary math:
[b_2 * x_2 = -b_0 - b_1 * x_1]
[x_2 = ]
So the intercept is
plot(xgrid,
col = c("red","blue")[as.numeric(ygrid)],
pch = 20,
cex = .2)
points(x,
col = y + 3,
pch = 19)
points(x[college_svm1$index,],
pch = 5,
cex = 2)
abline(-beta0 / beta[2], -beta[1]/beta[2])
The margins have the same slope as the decision boundary but a different intercept. Under the formulation used by the svm() function and described on page 357 of ISLR, the margin equals 1 (that’s the vertical distance between the decision boundary and the margin lines). Thus, we want the intercept of the lines:
[b_0 + b_1 * x_1 + b_2 * x_2 = ]
[x_2 = ]
[x_2 = - * x_1]
So, the intercepts are
plot(xgrid,
col = c("red","blue")[as.numeric(ygrid)],
pch = 20,
cex = .2)
points(x,
col = y + 3,
pch = 19)
points(x[college_svm1$index,],
pch = 5,
cex = 2)
abline((-1 - beta0) / beta[2], -beta[1] / beta[2], lty = 2)
abline((+1 - beta0) / beta[2], -beta[1] / beta[2], lty = 2)
Let’s see what happens if we decrease the cost of errors and repeat the same exercises. To make this easy to repeat for different cost values, we stick our code in a function.
<- function(df, cost_val, xgrid) {
svm_cost
<- svm(Private ~ Outstate + accept_rate,
svmfit data = df,
kernel = "linear",
cost = cost_val,
scale = FALSE)
print(paste("# of support vectors =", length(svmfit$index)))
<- drop(t(svmfit$coefs) %*% x[svmfit$index,])
beta <- -svmfit$rho
beta0 <- predict(svmfit, xgrid)
ygrid plot(xgrid, col = c("red","blue")[as.numeric(ygrid)], pch = 20, cex = .2)
points(x, col = y + 3, pch = 19)
points(x[df$index,], pch = 5, cex = 2)
abline(-beta0 / beta[2], -beta[1]/beta[2])
abline((-1 - beta0) / beta[2], -beta[1] / beta[2], lty = 2)
abline((+1 - beta0) / beta[2], -beta[1] / beta[2], lty = 2)
}
Run the function svm_cost() using cost parameters 10, 1, 0.1, 0.01
Example:
svm_cost(small_data, .01, xgrid)
[1] "# of support vectors = 57"
At first, the margins widen within the visible frame, but we can see that the decision boundary changes as we decrease the cost of errors. Additionally, as the cost of errors decrease, the number of support vectors increase.
While the visual exercise is useful in building intuition, often you will work with your full data set. In this case and in general, cross-validation is preferred to visual inspection. You can perform cross-validation to select among models using linear kernels by using the function tune()
set.seed(222)
<- tune(svm,
tune_linear ~ accept_rate + Outstate,
Private data = transformed_data,
kernel = "linear",
ranges = list(cost = c(0.001, 0.01, 0.1, 1, 5, 10, 100)))
We can see which cost had the best performance and compare how cross-validation errors varied across cost values. In the case of our data set, performance levels off after cost = 1.
summary(tune_linear)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost
0.1
- best performance: 0.1479687
- Detailed performance results:
cost error dispersion
1 1e-03 0.2728938 0.03581616
2 1e-02 0.2150516 0.04654521
3 1e-01 0.1479687 0.03534060
4 1e+00 0.1479687 0.03258152
5 5e+00 0.1479687 0.03085414
6 1e+01 0.1492674 0.03148925
7 1e+02 0.1492674 0.03148925
We can set the best model accordingly:
<- tune_linear$best.model
best_linear_mod summary(best_linear_mod)
Call:
best.tune(METHOD = svm, train.x = Private ~ accept_rate + Outstate,
data = transformed_data, ranges = list(cost = c(0.001, 0.01,
0.1, 1, 5, 10, 100)), kernel = "linear")
Parameters:
SVM-Type: C-classification
SVM-Kernel: linear
cost: 0.1
Number of Support Vectors: 328
( 164 164 )
Number of Classes: 2
Levels:
No Yes
As we’ve done with other models this semester, you could split your data into test and training data, train your model on the training data (including selecting a model) using cross-validation and then test its performance on the test data.
Remember that we used only two features. This was useful for visual inspection, but often you have access to many more features. What would happen if we ran the same model using all features available in the original data set college_data? Unsurprisingly, we can obtain much lower error with more features.
<- tune(svm,
tune_linear_full ~ .,
Private data = college_data,
kernel = "linear",
ranges = list(cost = c(0.001, 0.01, 0.1,
1, 5, 10, 100)))
summary(tune_linear_full)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost
5
- best performance: 0.05401265
- Detailed performance results:
cost error dispersion
1 1e-03 0.14673660 0.04776340
2 1e-02 0.07335997 0.03590371
3 1e-01 0.06565102 0.02678567
4 1e+00 0.05657676 0.02504778
5 5e+00 0.05401265 0.02162469
6 1e+01 0.05787546 0.02029151
7 1e+02 0.05915751 0.02023357
Mayve we think that a non-linear decision boundary would be better suited for our data. Thus, we would apply the support vector machine with a non-linear kernel. Some options include setting the kernel to “polynomial”, where we specify the degree d or “radial”, where we specify the positive constant gamma (larger values of gamma lead to more flexible models). Let’s start simple with a 3rd degree polynomial.
<- svm(Private ~ Outstate + accept_rate,
svm_poly3 data = small_data,
kernel = "polynomial",
degree = 3,
cost = 1)
plot(svm_poly3, small_data)
As before we can make a nice image of the model boundaries using the grid of x-values
<- predict(svm_poly3, xgrid)
ygrid plot(xgrid, col = c("red","blue")[as.numeric(ygrid)],
pch = 20, cex = .2)
points(x, col = y + 3, pch = 19)
To plot the decision boundary, we first use the predict() function with an additional argument decision.values = TRUE. This adds an attribute to the output, which allows you to recover the predicted value (not just predicted class) for each unique value in the test set (the x-grid in this case).
<- predict(svm_poly3, xgrid,
predicted_grid decision.values = TRUE)
<- attributes(predicted_grid)$decision predicted_grid
We can then plot the decision boundary using the contour() function with level = 0
<- predict(svm_poly3, xgrid)
ygrid plot(xgrid, col = c("red","blue")[as.numeric(ygrid)],
pch = 20, cex = .2)
points(x, col = y + 3, pch = 19)
contour(unique(xgrid[,1]), unique(xgrid[,2]),
matrix(predicted_grid, 75, 75),
level = 0,
add = TRUE)
As before, we can use the tune() function to see how different cost values compare. Try writing the tune function to test polynomials of degree 2 and 3 and with cost values of 0.001, 0.01, 1, 5, and 100:
set.seed(222)
<- tune(svm,
tune_poly ~ .,
Private data = transformed_data,
kernel = "polynomial",
ranges = list(cost = c(0.001, 0.1, 1, 5, 100),
degree = c(2, 3)))
summary(tune_poly)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost degree
100 3
- best performance: 0.1867299
- Detailed performance results:
cost degree error dispersion
1 1e-03 2 0.2728938 0.03581616
2 1e-01 2 0.2728938 0.03581616
3 1e+00 2 0.2728938 0.03581616
4 5e+00 2 0.2728938 0.03581616
5 1e+02 2 0.2728938 0.03581616
6 1e-03 3 0.2728938 0.03581616
7 1e-01 3 0.2098402 0.03360308
8 1e+00 3 0.1970196 0.04169226
9 5e+00 3 0.1879953 0.03498841
10 1e+02 3 0.1867299 0.03300264
We can also try with a radial kernel
set.seed(222)
<- tune(svm, Private ~ .,
tune_radial data = transformed_data,
kernel = "radial",
ranges = list(cost = c(0.001, 0.1, 1, 5, 100),
gamma = c(0.0001, 0.001, 0.01, 0.1, 1)))
summary(tune_radial)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost gamma
100 1
- best performance: 0.1428238
- Detailed performance results:
cost gamma error dispersion
1 1e-03 1e-04 0.2728938 0.03581616
2 1e-01 1e-04 0.2728938 0.03581616
3 1e+00 1e-04 0.2728938 0.03581616
4 5e+00 1e-04 0.2728938 0.03581616
5 1e+02 1e-04 0.1609058 0.03184748
6 1e-03 1e-03 0.2728938 0.03581616
7 1e-01 1e-03 0.2728938 0.03581616
8 1e+00 1e-03 0.2728938 0.03581616
9 5e+00 1e-03 0.2150516 0.04654521
10 1e+02 1e-03 0.1466866 0.03533624
11 1e-03 1e-02 0.2728938 0.03581616
12 1e-01 1e-02 0.2728938 0.03581616
13 1e+00 1e-02 0.1596237 0.03126915
14 5e+00 1e-02 0.1505495 0.03573142
15 1e+02 1e-02 0.1492674 0.03090384
16 1e-03 1e-01 0.2728938 0.03581616
17 1e-01 1e-01 0.1660839 0.03899615
18 1e+00 1e-01 0.1531136 0.03485918
19 5e+00 1e-01 0.1518149 0.03706845
20 1e+02 1e-01 0.1505328 0.03353007
21 1e-03 1e+00 0.2728938 0.03581616
22 1e-01 1e+00 0.1531136 0.03787237
23 1e+00 1e+00 0.1505495 0.03246168
24 5e+00 1e+00 0.1428738 0.02609063
25 1e+02 1e+00 0.1428238 0.02848941
Comparing cross-validation errors between the radial and polynomial fits, it seems that radial is better for our setting. Let’s understand how the radial decision boundary is drawn. Recall that after running tune() we can extract the best model
<- tune_radial$best.model
svm_radial
## Make predictions for the xgrid
<- predict(svm_radial, xgrid)
ygrid
## And plot the grid predictions and decision boundary
plot(xgrid, col = c("red","blue")[as.numeric(ygrid)],
pch = 20, cex = .2)
points(x, col = y + 3, pch = 19)
<- predict(svm_radial, xgrid, decision.values = TRUE)
predicted_grid <- attributes(predicted_grid)$decision
predicted_grid contour(unique(xgrid[,1]), unique(xgrid[,2]),
matrix(predicted_grid, 75, 75), level = 0, add = TRUE)
Again, we can compare our radial and polynomial models on the two-feature data to the same models on the original full data set college_data.
<- tune(svm,
full_poly ~ .,
Private data = college_data,
kernel = "polynomial",
ranges = list(cost = c(0.001, 0.1, 1, 5, 100),
degree = c(2, 3)))
summary(full_poly)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost degree
5 3
- best performance: 0.07462537
- Detailed performance results:
cost degree error dispersion
1 1e-03 2 0.27157842 0.04464526
2 1e-01 2 0.20079920 0.04546786
3 1e+00 2 0.16345321 0.04062089
4 5e+00 2 0.14029304 0.03357035
5 1e+02 2 0.14157509 0.03847015
6 1e-03 3 0.25872461 0.03991465
7 1e-01 3 0.16861472 0.05260691
8 1e+00 3 0.08618049 0.03520954
9 5e+00 3 0.07462537 0.03297477
10 1e+02 3 0.09522145 0.02911267
summary(tune_poly)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost degree
100 3
- best performance: 0.1867299
- Detailed performance results:
cost degree error dispersion
1 1e-03 2 0.2728938 0.03581616
2 1e-01 2 0.2728938 0.03581616
3 1e+00 2 0.2728938 0.03581616
4 5e+00 2 0.2728938 0.03581616
5 1e+02 2 0.2728938 0.03581616
6 1e-03 3 0.2728938 0.03581616
7 1e-01 3 0.2098402 0.03360308
8 1e+00 3 0.1970196 0.04169226
9 5e+00 3 0.1879953 0.03498841
10 1e+02 3 0.1867299 0.03300264
<- tune(svm,
full_radial ~ .,
Private data = college_data,
kernel = "radial",
ranges = list(cost = c(0.001, 0.1, 1, 5, 100),
gamma = c(0.0001, 0.001, 0.01, 0.1, 1)))
summary(full_radial)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost gamma
100 0.01
- best performance: 0.05656011
- Detailed performance results:
cost gamma error dispersion
1 1e-03 1e-04 0.27272727 0.05678760
2 1e-01 1e-04 0.27272727 0.05678760
3 1e+00 1e-04 0.27144522 0.05669836
4 5e+00 1e-04 0.14400599 0.05642052
5 1e+02 1e-04 0.06939727 0.03735527
6 1e-03 1e-03 0.27272727 0.05678760
7 1e-01 1e-03 0.27144522 0.05669836
8 1e+00 1e-03 0.10541126 0.04419069
9 5e+00 1e-03 0.06811522 0.03728823
10 1e+02 1e-03 0.06298701 0.02661123
11 1e-03 1e-02 0.27272727 0.05678760
12 1e-01 1e-02 0.11313686 0.04576357
13 1e+00 1e-02 0.06683317 0.03566760
14 5e+00 1e-02 0.06040626 0.02900692
15 1e+02 1e-02 0.05656011 0.02099165
16 1e-03 1e-01 0.27272727 0.05678760
17 1e-01 1e-01 0.10929071 0.04864422
18 1e+00 1e-01 0.06683317 0.02937271
19 5e+00 1e-01 0.07069597 0.03021775
20 1e+02 1e-01 0.07069597 0.03471769
21 1e-03 1e+00 0.27272727 0.05678760
22 1e-01 1e+00 0.27272727 0.05678760
23 1e+00 1e+00 0.26500167 0.06185452
24 5e+00 1e+00 0.24831835 0.05311922
25 1e+02 1e+00 0.24831835 0.05311922
summary(tune_radial)
Parameter tuning of 'svm':
- sampling method: 10-fold cross validation
- best parameters:
cost gamma
100 1
- best performance: 0.1428238
- Detailed performance results:
cost gamma error dispersion
1 1e-03 1e-04 0.2728938 0.03581616
2 1e-01 1e-04 0.2728938 0.03581616
3 1e+00 1e-04 0.2728938 0.03581616
4 5e+00 1e-04 0.2728938 0.03581616
5 1e+02 1e-04 0.1609058 0.03184748
6 1e-03 1e-03 0.2728938 0.03581616
7 1e-01 1e-03 0.2728938 0.03581616
8 1e+00 1e-03 0.2728938 0.03581616
9 5e+00 1e-03 0.2150516 0.04654521
10 1e+02 1e-03 0.1466866 0.03533624
11 1e-03 1e-02 0.2728938 0.03581616
12 1e-01 1e-02 0.2728938 0.03581616
13 1e+00 1e-02 0.1596237 0.03126915
14 5e+00 1e-02 0.1505495 0.03573142
15 1e+02 1e-02 0.1492674 0.03090384
16 1e-03 1e-01 0.2728938 0.03581616
17 1e-01 1e-01 0.1660839 0.03899615
18 1e+00 1e-01 0.1531136 0.03485918
19 5e+00 1e-01 0.1518149 0.03706845
20 1e+02 1e-01 0.1505328 0.03353007
21 1e-03 1e+00 0.2728938 0.03581616
22 1e-01 1e+00 0.1531136 0.03787237
23 1e+00 1e+00 0.1505495 0.03246168
24 5e+00 1e+00 0.1428738 0.02609063
25 1e+02 1e+00 0.1428238 0.02848941